home *** CD-ROM | disk | FTP | other *** search
/ MacWorld 1998 March / Macworld (1998-03) (Disk 1).dmg / Shareware World / Utilities / Text Processing / Alpha / Tcl / Modes / Perl Mode / perlMode.tcl < prev    next >
Encoding:
Text File  |  1997-12-10  |  45.8 KB  |  1,588 lines  |  [TEXT/ALFA]

  1. #############################################################################
  2. # perlMode.tcl
  3. # -----------
  4. #
  5. # This is a set of routines that allow Alpha to act as a front end for the
  6. # standalone MacPerl application and that allow Perl scripts to be used as 
  7. # text filters in Alpha.  These functions are accessed through a special 
  8. # MacPerl menu.
  9. #
  10. # The features of this package are explained in the file "MacPerl Help",
  11. # accessible from the Help menu.
  12. #
  13. #############################################################################
  14. #
  15. # If you don't already have MacPerl, it's available by anonymous ftp from
  16. # the umich site
  17. #
  18. #   mac.archive.umich.edu    [141.211.165.34]    mac/development/languages
  19. #
  20. # and its mirrors.  Also, MacPerl's home site is 
  21. #
  22. #   ftp.switch.ch            [130.59.1.40]        software/mac/src/mpw_c
  23. #
  24. # MacPerl was written (ported to the Mac) by 
  25. #        Matthias Neeracher <neeri@iis.ee.ethz.ch> , and
  26. #        Tim Endres <time@ice.com>.
  27. #
  28. #############################################################################
  29. # Author: Tom Pollard <pollard@schrodinger.com>
  30. #
  31. # Contributors: Dan Herron     <herron@cogsci.ucsd.edu>
  32. #               David Schooley <schooley@ee.gatech.edu>
  33. #               Vince Darley   <darley@fas.harvard.edu>
  34. #               Martijn Koster <m.koster@nexor.co.uk>
  35. #
  36. # Version History:
  37. #
  38. # 3.12 10/97 -  Uses new menu-building code, so you can add with menu::insert (v)
  39. # 3.11 9/97  -  Fixed problem with modevars in new Alpha scheme (Johan)
  40. # 3.10 8/97  -  Modernised for new Alpha Tcl scheme (vince)
  41. # 3.0  4/97  -  MacPerl interactions don't depend on MacPerl app name anymore
  42. #               Fixed bug with perlFileAsFilter ($scriptStart uninitialized)
  43. # 2.9  3/97  -  Fixed bug in command-dbl-click help lookup for Perl5 mode
  44. # 2.8  2/97  -  Added Quick-Save commands in new submenu [Dan Herron]
  45. #               "Save As CGI" finally works.
  46. # 2.7  2/97  -  Comments before "#!/bin/perl" no longer confuse 'gotoPerlError'
  47. # 2.6  2/97  -  Added electricPerlLeft and electricPerlRight - [David Schooley]
  48. # 2.51 1/96  -  Fixed problem w/ "Tell MacPerl:Save As..."
  49. # 2.5  1/96  -  Colorization and cmd-dbl-click modified to support Perl 5 docs
  50. # 2.41 7/95  -  Minor tweaks
  51. # 2.4  7/95  -  Fixed bugs affecting running unsaved scripts and error handling
  52. # 2.3  7/95  -  Minor tweaks and code rearrangement.
  53. # 2.2  6/95  -  Text filters act only on current line if "Apply to Buffer" is
  54. #                  false and no text has been selected.
  55. #               Bug fix in error-marking for scripts sent as AppleEvent params.
  56. #               Cmd-dbl-clicking a function call jumps to function, if
  57. #                  defined in the same file.
  58. # 2.1  6/95  -  Cmd-dbl-clicking a 'require'd filename opens the file.
  59. # 2.0  6/95  -  Minor bug fixes (incl. keyword decapitalization)
  60. #               Alpha 6.0b17 compatibility updates.
  61. #               Text Filters folder is settable from the App Paths menu now.
  62. # 1.9  5/95  -  Cmd-dbl-clicking Perl keywords and special variables displays
  63. #                  the man page info.
  64. # 1.81 4/95  -  one very minor Alpha compatibility update (winInfo->getWinInfo).
  65. # 1.8  4/95  -  Menu reorganized somewhat.
  66. #               Text Filters folder can now be anywhere.
  67. #               "ApplyToBuffer" flag ignored if text has been selected.
  68. #               Bug fixes.
  69. # 1.7  1/95  -  Updated to take advantage of MacPerl 4.1.4 AppleEvent features:
  70. #                1) Text filters use 'batch' doScript (.: STDOUT file obsolete)
  71. #                2) Filter scripts sent as doScript params (.: SCRIPT file obsolete)
  72. #                3) "Save As Droplet" and "Save as Runtime" commands added.
  73. #               Errors generated in 'require'd files are now displayed correctly
  74. # 1.6 10/94  -  "UseDebugger" flag added (forces scripts to run under debugger).
  75. #               Key bindings added for some menu commands.
  76. #               "perlDoScript{,2,3}" procs consolidated into a single proc.
  77. #               "saveAndRun" option added.
  78. #               Command-line args now parsed into units more correctly, in
  79. #                   particular, quoted file names aren't broken up.
  80. #               "Close Output Window" added to "Tell MacPerl" menu.
  81. #               Updated for Alpha 5.98 to load when menu is inserted.
  82. #               The error messages window is now recycled.
  83. #               "perlRecycleOutput" recycles output window.
  84. #               Minor bug fixes.
  85. # 1.5  9/94  -  MacPerl menu rearranged somewhat.
  86. #               Explicit "Get Output Window" command added to menu.
  87. #               Reading "#!" line for args is incompatible w/ standard,
  88. #                   so it's been dropped.
  89. #               Only scan the first 40 output lines for error messages (faster)
  90. #                "wrapFilterScript" no longer opens STDIN
  91. #               Text filters may now use command-line args
  92. #               STDIN for text filters passed as explicit cmd-line arg 
  93. # 1.4  9/94  -  The "#!" line of every script is read for command-line args,
  94. #                    which are passed explicitly to MacPerl with the script.
  95. #                "PromptForArgs" menu flag added.
  96. #                "perlCmdlineArgs" modeVar holds default command-line args.
  97. #                Scripts are sent using custom "perlDoScript2" proc, which
  98. #                    allows passing of explicit command-line args.
  99. # 1.3  9/94  -  When any script generates a compilation error, the file 
  100. #                    containing the script is brought up with the offending 
  101. #                    line highlighted; all error output is also written to
  102. #                    a "Perl Error Messages" window.
  103. #                'repeatLastFilter' runs again the last text-filter script used.
  104. #                'perlLastFilter' modeVar holds pathname of last filter.
  105. #                Menu flags now mirrored as modeVars, so they can be saved and
  106. #                    restored between sessions.
  107. #                Minor bug fixes.
  108. # 1.2  8/94  -  'retrieveOutput' and 'autoSwitch' flags added.
  109. #                'openInMacperl' added.
  110. #                MacPerl output window now closed before new scripts are sent.
  111. #                Filters now abort if there are compilation errors, and
  112. #                MacPerl diagnostic output retrieved and displayed in Alpha.
  113. # 1.1  8/94  -  'quitMacperl' added.
  114. #               perl-mode file-marking updated for Alpha 5.90
  115. #               Simplified installation via 'loadMacperl'(Pete Keleher). 
  116. # 1.0  7/94  -  perl-mode setup updated for Alpha 5.85:
  117. #                    keyword colorization supported
  118. #                    custom file-marking added
  119. #               #! lines in filter scripts now handled correctly 
  120. #               Workarounds installed for AppleEvent bug in MacPerl 4.1.3
  121. # 0.9  3/94  -  perl-mode stuff added, and
  122. #               highlighted 'Perl commands' file (man page) prepared
  123. #               minor bug fixes, too
  124. # 0.8  3/94  -  flags are now check-marked
  125. # 0.7  3/94  -  nested Text Filters folder now supported
  126. #               menu format modified somewhat
  127. # 0.6  3/94  -  'applyToBuffer' flag added
  128. #               scripts in Alpha buffers can now be used as filters 
  129. # 0.5  2/94  -  'filters', 'open special' submenu added
  130. #               'overwrite' flag added
  131. # 0.2  1/94  -  menu support added (Martijn Koster <m.koster@nexor.co.uk>)
  132. #               'execute selection', 'execute buffer' commands added
  133. # 0.1  9/93  -  text filter functionality created
  134. #                  
  135. ##############################################################################
  136. #
  137. alpha::mode Perl 3.11 perlMenu {*.pl *.ph *.pm} perlMenu {
  138.     addMenu perlMenu "•132"
  139. } help {file "MacPerl Help"} uninstall {this-directory}
  140.  
  141. proc dummyPerl {} {}
  142.  
  143. #############################################################################
  144. #  Default settings for the Perl menu flags  
  145. #
  146. newPref f perluseDebugger 0 Perl shadowPerl
  147. newPref f perlretrieveOutput 1 Perl shadowPerl
  148. newPref f perlautoSwitch 1 Perl shadowPerl
  149. newPref f perloverwriteSelection 0 Perl shadowPerl
  150. newPref f perlapplyToBuffer 1 Perl shadowPerl
  151. newPref f perlpromptForArgs 0 Perl shadowPerl
  152. newPref f perlRecycleOutput 0 Perl
  153. newPref v perlPrevScript {*startup*} Perl
  154. newPref v perlCmdlineArgs {} Perl
  155. newPref v perlVersion {5} Perl shadowPerl [list 4 5]
  156.  
  157. newPref v perlFilterPath "$HOME:Tcl:Packages:Text Filters:" Perl rebuildFilterMenu
  158. newPref v perlLibFolder "" Perl buildPerlSearchPath
  159.  
  160. # Perl mode relies on the old Alpha scheme that there is both a 
  161. # PerlmodeVars(something) and a variable 'something'. This is the easiset 
  162. # way to fix this. It would be better to rewrite Perl mode to only use its
  163. # modevar array, but if I do that I will probably introduce a large number
  164. # of bugs. 
  165. # -- Johan
  166. foreach __var [list perluseDebugger perlretrieveOutput perlautoSwitch perloverwriteSelection \
  167.   perlapplyToBuffer perlpromptForArgs perlRecycleOutput perlPrevScript perlCmdlineArgs \
  168.   perlVersion perlFilterPath perlLibFolder] {
  169.     set $__var $PerlmodeVars($__var)
  170. }
  171. unset __var
  172.  
  173. #############################################################################
  174. # Other Perl-mode variable definitions
  175. #
  176. newPref f elecRBrace        {0} Perl
  177. newPref f elecLBrace        {1} Perl
  178. newPref f electricSemi    {0} Perl
  179. newPref f electricTab        {1} Perl
  180. newPref f electricReturn    {1} Perl
  181. newPref v wordBreak        {[$%@*]?\w+} Perl
  182. newPref v prefixString    {# } Perl
  183. newPref f wordWrap        {0} Perl
  184. newPref v funcExpr        {^sub *([+-a-zA-Z0-9]+)} Perl
  185. newPref v wordBreakPreface        {[^a-zA-Z0-9_%@*\$]} Perl
  186. newPref f autoMark    1    Perl
  187. newPref v stringColor    green    Perl
  188.  
  189. # ALL THE ABOVE VARS ARE NOW GLOBAL AND MODE-VARS
  190.  
  191. ##############################################################################
  192. # Miscellaneous definitions
  193. #
  194. set perlErrorWindow {* Perl Error Messages *}
  195. set perlOutputWindow {* Perl Output *}
  196. set interpPat {(#![     !-~]*)}
  197.  
  198. set perlFilterMenu "textFilters"
  199.  
  200. if {[catch "perl${perlVersion}.tcl"]} {
  201.     alertnote "Couldn't load the Perl-mode colorization file \"perl${perlVersion}.tcl\".  Contact the maintainer."
  202. }
  203.  
  204. #############################################################################
  205. #  Return paths to standard files, based on the path to MacPerl:
  206. #
  207. proc macperlFolder {} {
  208.    set name [nameFromAppl McPL]
  209.    regexp {(.*):([^:]*)} $name pathname dirname filename
  210.    return ${dirname}:
  211. }
  212.  
  213. proc stdinPath {} {
  214.    return [macperlFolder]STDIN
  215. }
  216.  
  217. proc scriptPath {} {
  218.    return [macperlFolder]SCRIPT
  219. }
  220.  
  221. #############################################################################
  222. # Define the dummy proc that will be called when the perl menu
  223. # is first inserted into the menubar
  224. #
  225. proc perlMenu {} {}
  226.  
  227. #############################################################################
  228. #  Build the perl menu
  229. #            
  230.  
  231. menu::buildProc perlMenu menu::buildPerl
  232. menu::buildProc generalOptions menu::buildgeneralOptions
  233. menu::buildProc filterOptions menu::buildfilterOptions
  234. menu::buildProc perlFilterMenu rebuildFilterMenu
  235.  
  236. proc menu::buildPerl {} {
  237.     global perlFilterMenu perlMenu perlPrevScript
  238.     set ma {
  239.         "/'<Umacperl"
  240.         {menu -m -n "tellMacperl..." -p perlTellProc {
  241.            "/O<UOpen This File"
  242.            "Save As Droplet"
  243.            "Save As Runtime"
  244.            "Save As CGI"
  245.             "(-"
  246.            "Get Output Window"
  247.            "Close Output Window"
  248.            "Quit"
  249.            }
  250.         } 
  251.         {menu -m -n "Quick Save As..." -p perlSaveProc {
  252.            "Droplet"
  253.            "Runtime"
  254.            "CGI"
  255.            }
  256.         } 
  257.         {menu -m -n help -p perlHelpProc {
  258.             "MacPerl Mode"
  259.             "Mac Specifics"
  260.             "Perl4 Manual"
  261.             "Perl5 Manual"
  262.         }}
  263.         "(-"
  264.         "runTheSelection"
  265.         "/R<UrunTheBuffer"
  266.         "/R<B<OsaveAndRun"
  267.         "runAFile"
  268.         "(-"
  269.     }
  270.     lappend ma [list menu -n $perlFilterMenu {}] \
  271.       "selectBufferAsFilter" "selectFileAsFilter"
  272.     if {$perlPrevScript == {} || $perlPrevScript == {*startup*}} {
  273.         lappend ma "/F<U(repeatLastFilter"
  274.     } else {
  275.         lappend ma "/F<UrepeatLastFilter"
  276.     }
  277.     lappend ma "(-" \
  278.       [list menu -n generalOptions {}] \
  279.       [list menu -n filterOptions {}]
  280.  
  281.     return [list build $ma -1 \
  282.       {generalOptions filterOptions perlFilterMenu} $perlMenu]
  283. }
  284.  
  285.  
  286. # General Perl-menu options menu
  287. #
  288. proc menu::buildgeneralOptions {} {
  289.     foreach i {"retrieveOutput" "autoSwitch" "promptForArgs" "useDebugger"} {
  290.         global perl$i
  291.         if [set perl$i] {
  292.             lappend ma "!•$i"
  293.         } else {
  294.             lappend ma $i
  295.         }
  296.     }
  297.     return [list build $ma]
  298. }
  299.  
  300. # Text Filter options menu
  301. #
  302. proc menu::buildfilterOptions {} {
  303.     uplevel \#0 {
  304.     menu -n filterOptions {
  305.         "applyToBuffer"
  306.         "overwriteSelection"
  307.         "(-"
  308.         "rebuildFilterMenu"
  309.     }    
  310.     markMenuItem filterOptions overwriteSelection $perloverwriteSelection
  311.     markMenuItem filterOptions applyToBuffer $perlapplyToBuffer
  312. }
  313.  
  314. }
  315.  
  316. #############################################################################
  317. #  Build a submenu of "preattached" Perl filters using the names of the 
  318. #  scripts in the Text Filters directory.  Called whenever Text Filters
  319. # folder is reassigned.
  320. #
  321. proc rebuildFilterMenu {{args}} {
  322.     global perlFilters perlFilterMenu perlFilterPath
  323.     global $perlFilterMenu
  324.     
  325.     eval [buildSubMenu [list $perlFilterPath] $perlFilterMenu textFiltersProc perlFilters]
  326. }
  327.  
  328. menu::buildSome perlMenu
  329.  
  330. # ShadowPerl sets the global vars when the mode vars are modified and
  331. # keeps the menu checkmarked correctly.
  332. #
  333. proc shadowPerl {name} {
  334.     global HOME perlMenu 
  335.     global perloverwriteSelection perlapplyToBuffer perlretrieveOutput perlautoSwitch
  336.     global perlpromptForArgs perlPrevScript perlCmdlineArgs perluseDebugger
  337.     switch $name {
  338.         "perluseDebugger"    {
  339.             markMenuItem generalOptions useDebugger $perluseDebugger
  340.          }
  341.         "perloverwriteSelection"    {
  342.             markMenuItem filterOptions overwriteSelection $perloverwriteSelection
  343.          }
  344.         "perlapplyToBuffer"    {
  345.             markMenuItem filterOptions applyToBuffer $perlapplyToBuffer
  346.          }
  347.         "perlretrieveOutput"    {
  348.             markMenuItem generalOptions retrieveOutput $perlretrieveOutput 
  349.         }
  350.         "perlautoSwitch" {    
  351.             markMenuItem generalOptions autoSwitch $perlautoSwitch 
  352.         }
  353.         "perlpromptForArgs" {    
  354.             markMenuItem generalOptions promptForArgs $perlpromptForArgs 
  355.         }
  356.         "perlVersion" {    
  357.             set modeCode "perl${perlVersion}.tcl"
  358.             if {[catch "$modeCode"]} {
  359.                 alertnote "Couldn't load the Perl-mode colorization file \"$modeCode\".  Contact the maintainer."
  360.             }
  361.         }
  362.         "perlLastFilter" {    
  363.             # Don't allow perlPrevScript to be changed from the flags menu
  364.             if {$perlPrevScript == "*startup*"} {
  365.                 set perlPrevScript $perlLastFilter
  366.                 enableMenuItem $perlMenu repeatLastFilter 1
  367.             } else {
  368.                 set perlLastFilter $perlPrevScript 
  369.             }
  370.         }
  371.     }
  372. }
  373.  
  374. #############################################################################
  375. # Menu commands
  376. #############################################################################
  377.  
  378. ############################################################################
  379. # Toggle the perl menu flags
  380. #
  381. proc retrieveOutput {} {
  382.     perlFlip perlretrieveOutput
  383. }
  384.  
  385. proc useDebugger {} {
  386.     perlFlip perluseDebugger
  387. }
  388.  
  389. proc autoSwitch {} {
  390.     perlFlip perlautoSwitch
  391. }
  392.  
  393. proc perlFlip {var} {
  394.     global $var
  395.     set $var [expr [set $var] ? 0 : 1]
  396.     synchroniseModeVar $var
  397.     shadowPerl $var
  398. }
  399.  
  400. proc overwriteSelection {} {
  401.     perlFlip perloverwriteSelection
  402. }
  403.  
  404. proc applyToBuffer {} {
  405.     perlFlip perlapplyToBuffer
  406. }
  407.  
  408. proc promptForArgs {} {
  409.     perlFlip perlpromptForArgs
  410. }
  411.  
  412. #############################################################################
  413. # Switch to MacPerl:
  414. proc macperl {} {
  415.     app::launchFore McPL
  416. }
  417.  
  418. #############################################################################
  419. # Interact with MacPerl in some other way besides executing a script
  420. #
  421. #DTH: note addition of two lines for auto-save
  422. proc perlTellProc {menu name} {
  423.     switch -exact $name {
  424.     "Open This File"        { openInMacperl }
  425.     
  426.     "Save As Droplet"        { saveThruMacperl "droplet" }
  427.     
  428.     "Save As Runtime"        { saveThruMacperl "runtime" }
  429.     
  430.     "Save As CGI"            { saveThruMacperl "cgi" }
  431.     
  432.     "Get Output Window"        { openPerlOutput }
  433.     
  434.     "Close Output Window"    { sendCloseWinName MacPerl $perlName ;
  435.                               sendCloseWinName MacPerl "Perl Debug" }
  436.                             
  437.     "Quit"                    { quitMacperl }
  438.     }
  439. }
  440.  
  441. proc perlSaveProc {menu name} {
  442.     switch -exact $name {
  443.     "Droplet"    { saveThruMacperl "auto-droplet" }
  444.     
  445.     "Runtime"    { saveThruMacperl "auto-runtime" }
  446.  
  447.     "CGI"        { saveThruMacperl "auto-cgi" }
  448.     }
  449. }
  450.  
  451. #############################################################################
  452. # Open the current file under MacPerl.  This used to useful for saving files 
  453. # as droplets or runtime scripts.  Maybe it's still useful for something...?
  454. #
  455. proc openInMacperl {} {
  456.     if {[winDirty]} {
  457.         case [askyesno -c "Save '[lindex [winNames] 0]'?"] in {
  458.             "yes" {save}
  459.             "no" {}
  460.             "cancel" {return}
  461.         }
  462.     }
  463.     set name [app::launchFore McPL]
  464.     sendOpenEvent -n [file tail $name] [win::Current]
  465. }
  466.  
  467. #############################################################################
  468. # Save the script in the current window as a MacPerl droplet or 
  469. # runtime script.  
  470. #
  471. proc saveThruMacperl {type} {
  472.     global ALPHA
  473.  
  474.     set name [file tail [app::launchBack McPL]]
  475.     getWinInfo arr
  476.     if {$arr(dirty) == 1} {
  477.         case [askyesno -c "Save '[lindex [winNames] 0]' source file also?"] in {
  478.             "yes" {save}
  479.             "no" {}
  480.             "cancel" {return}
  481.         }
  482.     }
  483.     #DTH note the following "if" block which replaced what is in the new "else" block
  484.     set myName [lindex [winNames -f] 0]
  485.     if {$type == "auto-droplet" || $type == "auto-runtime"} {
  486.         if {[file extension $myName] == ".pl"} {
  487.             set destfile [AEFilename [file rootname $myName]]
  488.         } else {
  489.             set destfile [AEFilename [file rootname $myName]]
  490.         }
  491.     } elseif {$type == "auto-cgi"} {
  492.         set destfile [AEFilename "[file rootname $myName].cgi"]
  493.     } else {
  494.         set destfile [AEFilename [putfile {Save droplet as} [lindex [winNames] 0]]]
  495.     }
  496.  
  497.     set script [curlyq [getText 0 [maxPos]]]
  498.     #DTH note addition of "auto-xxx" in two lines below
  499.     if {$type == "droplet" || $type == "auto-droplet"} {
  500.         set saveType "SCPT"
  501.     } elseif {$type == "runtime" || $type == "auto-runtime"} {
  502.         set saveType "MrP7"
  503.     } elseif {$type == "cgi" || $type == "auto-cgi"} {
  504.         set saveType "'WWWΩ'"
  505.     } elseif {$type == "text"} {
  506.         set saveType "TEXT"
  507.     }
  508.     
  509.     set err [catch {eval "AEBuild -t 36000 -r \"$name\"" core save {----} [list $script] {dest:} [list $destfile] {fltp:} $saveType } reply ]
  510.     if {$err} { message "AEBuild error code $err in saveThruMacperl" }
  511.     
  512. # The following lines could be used to tell MacPerl to take the script file 
  513. # from an existing disk file and then re-save it in the desired form.
  514. #
  515. #    set srcfile "\[ [AEFilename [win::Current]] \]"
  516. #    set reply [eval "AEBuild -t 36000 -r \"$name\"" core save {----} [list $srcfile] {dest:} [list $destfile] {fltp:} $saveType ]
  517. #
  518. }
  519.  
  520. #############################################################################
  521. # Quit a running MacPerl app:
  522. proc quitMacperl {} {
  523.     foreach proc [processes] {
  524.         set sig [lindex $proc 1]
  525.         if {$sig == "McPL"} {
  526.             sendQuitEvent [lindex $proc 0]
  527.             # switchTo is necessary to keep MacPerl from blinking
  528.             switchTo [lindex $proc 0]    
  529.         }
  530.     }
  531. }
  532.  
  533. #############################################################################
  534. # Run the selection as a MacPerl script:
  535. # (No special arrangements are made to provide input or capture the output)
  536. proc runTheSelection {} {
  537.     global scriptFile scriptStart
  538.     set scriptFile [win::Current]
  539.     set scriptStart [lindex [posToRowCol [getPos]] 0]
  540.     perlExecuteScript [getSelect]
  541. }
  542.  
  543. proc runTheBuffer {} {
  544.     global scriptFile scriptStart
  545.     set scriptFile [win::Current]
  546.     set scriptStart 1
  547.     perlExecuteScript [getText 0 [maxPos]]
  548. }
  549.  
  550. proc runAFile {} {
  551.     global scriptFile scriptStart
  552.     if {! [catch {getfile "Select a Perl script"} path]} {
  553.         set scriptFile $path
  554.         set scriptStart 1
  555.         perlExecuteFile $path
  556.     }
  557. }
  558.  
  559. proc saveAndRun {} {
  560.     global scriptFile scriptStart
  561.     save
  562.     set path [win::Current]   
  563.     set scriptFile $path
  564.     set scriptStart 1
  565.     perlExecuteFile $path
  566. }
  567.  
  568. #############################################################################
  569. # Run a preattached Perl text-filter script selected from the menu:
  570. #
  571. proc textFiltersProc {menu name} {
  572.     global perlFilters scriptFile scriptStart
  573.     
  574.     perlFileAsFilter $perlFilters($menu:$name)
  575. }
  576.  
  577. #############################################################################
  578. # Reuse the previous (buffer or file) filter:
  579. #
  580. proc repeatLastFilter {} {
  581.     global scriptFile scriptStart perlPrevScript perlMenu 
  582.     if {$perlPrevScript != {}} {
  583.         set stype [lindex $perlPrevScript 0]
  584.         set name [lindex $perlPrevScript 1]
  585.         if {$stype == "file"} {
  586.             perlFileAsFilter $name
  587.         } elseif {$stype == "buffer"} {
  588.             perlBufferAsFilter $name
  589.         } else {
  590.             message "Bogus filter name : \"$perlPrevScript\""
  591.             set perlPrevScript {}
  592.             synchroniseModeVar perlLastFilter $perlPrevScript
  593.             enableMenuItem $perlMenu repeatLastFilter 0
  594.         }
  595.     }
  596. }
  597.  
  598. #############################################################################
  599. # Ask for a file containing a Perl script to use as a filter:
  600. #
  601. proc selectFileAsFilter {} {
  602.     global scriptFile scriptStart perlPrevScript
  603.     if {! [catch {getfile "Select a MacPerl script"} path]} {
  604.         perlFileAsFilter $path
  605.     }
  606. }
  607.  
  608. #############################################################################
  609. # Ask for an Alpha buffer containing a Perl script to use as a filter:
  610. #
  611. proc selectBufferAsFilter {} {
  612.     global scriptFile scriptStart perlPrevScript
  613.     
  614.     set windows [winNames]
  615.     set current [lindex $windows 0]
  616.     if {[llength $windows] > 1} {
  617.         set name [listpick [lsort $windows]]
  618.         if {[string length $name]} {
  619.             # get the full name of the chosen window
  620.             set wname [lindex [winNames -f] [lsearch -exact $windows $name]]
  621.             perlBufferAsFilter $wname
  622.            }
  623.     }
  624. }
  625.  
  626. #############################################################################
  627. # Open a file from the MacPerl application folder - used by "Open Special"
  628. #
  629. proc perlOpenFile {menu name} {
  630.     set filename [macperlFolder]$name
  631.     if {[file exists $filename]} {
  632.         edit $filename
  633.     } else {
  634.         alertnote "That file doesn't exist yet"
  635.     }
  636. }
  637.  
  638. #############################################################################
  639. # Support procs
  640. #############################################################################
  641.  
  642. #############################################################################
  643. # Prompt the user to enter a string containing command-line args.
  644. #
  645. proc getCmdlineArgs {} {
  646.     global perlCmdlineArgs
  647.     if {![catch {prompt "Command-line arguments (if any):" $perlCmdlineArgs} args]} {
  648.         synchroniseModeVar perlCmdlineArgs $args
  649.     } else {
  650.         error "getCmdlineArgs: User cancelled"
  651.     }
  652.     return $args
  653. }
  654.  
  655. #############################################################################
  656. # Tell MacPerl to run a script file:
  657. #
  658. proc perlExecuteFile {path {args {}} {flags {}}} {
  659.     global ALPHA
  660.     global perlretrieveOutput perlautoSwitch perlpromptForArgs perluseDebugger
  661.     global scriptFile scriptStart filterHeadLen perlName
  662.     
  663.     if {[string length $path]} {
  664.         set perlName [file tail [app::launchBack McPL]]
  665.         if {[string length $perlName]} {
  666.                 
  667.             set ok [regexp {(.*):([^:]*)} $path pathname dirname filename]
  668.             if {!$ok} {    set name $wname    }
  669.  
  670.             if {$path != [scriptPath]} {    
  671.                 set filterHeadLen 0    
  672.             }
  673.             
  674.             if {$perluseDebugger} {
  675.                 append flags "debug"
  676.             }
  677.             if {$perlpromptForArgs} { 
  678.                 append args " [getCmdlineArgs]"
  679.             }
  680.             
  681.             sendCloseWinName $perlName $perlName
  682.             sendCloseWinName $perlName "Perl Debug"
  683.             if {$perlautoSwitch || $perluseDebugger} {
  684.                 switchTo $perlName
  685.             } else {
  686.                 message "Running file \"$filename\" as Perl script"
  687.                 watchCursor
  688.             }
  689.             
  690.             perlDoScript $perlName $path $args {} $flags
  691.             
  692. # (not sure which choice is better...)
  693. #            if {!$perlautoSwitch} {switchTo $ALPHA}
  694.             switchTo $ALPHA
  695. #
  696.             if {![getMacPerlError]} {
  697.                 if {$perlretrieveOutput} {openPerlOutput}
  698.             }
  699.         } else {
  700.             alertnote "Couldn't run MacPerl"
  701.         }
  702.     } else {
  703.         alertnote "No file specified to execute"
  704.     }
  705. }
  706.  
  707. #############################################################################
  708. # Run a MacPerl script, passed explicitly as a string:
  709. #
  710. # If no "#!/bin/perl" line already exists, one is preprended to the script
  711. # by wrapSelectScript, which also sets $filterHeadLen for use by 
  712. # getMacPerlError.
  713. proc perlExecuteScript {script {args ""} {flags {}} } {
  714.     global perlretrieveOutput perlautoSwitch perlpromptForArgs perlName
  715.     global scriptFile scriptStart filterHeadLen perluseDebugger ALPHA
  716.     
  717.     if {$script != ""} {
  718.         set script [wrapSelectScript $script]
  719.         
  720.         if {![regexp {(.*):([^:]*)} $scriptFile pathname dirname filename]} {
  721.             set filename $scriptFile 
  722.         }
  723.  
  724.         set perlName [file tail [app::launchBack McPL]]
  725.         if {[string length $perlName]} {
  726.         
  727.             if {$perluseDebugger} {
  728.                 append flags "debug"
  729.             }
  730.             if {$perlpromptForArgs} { 
  731.                 append args " [getCmdlineArgs]"
  732.             }
  733.             
  734.             sendCloseWinName $perlName $perlName
  735.             sendCloseWinName $perlName "Perl Debug"
  736.             if {$perlautoSwitch || $perluseDebugger} {
  737.                 switchTo $perlName
  738.             } else {
  739.                 message "Running buffer \"$filename\" as Perl script"
  740.                 watchCursor
  741.             }
  742.             
  743.             perlDoScript $perlName $script $args {} $flags
  744.             
  745.             switchTo $ALPHA
  746.  
  747.             if {![getMacPerlError]} {
  748.                 if {$perlretrieveOutput} {openPerlOutput}
  749.             }
  750.         }
  751.         
  752.     } else {
  753.             alertnote "Can't run an empty script"
  754.     }
  755. }
  756.  
  757. #############################################################################
  758. # Prepare the contents of a disk file for use as a text-filter script. 
  759. # (calls perlTextFilter to actually run the script)
  760. proc perlFileAsFilter {path} {
  761.     global scriptFile scriptStart perlPrevScript perlMenu 
  762.     
  763.     regexp {(.*):([^:]*)} $path pathname dirname name
  764.     
  765.     if {![catch {readFile $path} coreScript]} {
  766.         set scriptFile $path
  767.         set scriptStart 1
  768.         set script [wrapFilterScript $coreScript]
  769.         set perlPrevScript [list "file" $path]
  770.         synchroniseModeVar perlLastFilter $perlPrevScript 
  771.         enableMenuItem $perlMenu repeatLastFilter 1
  772.         message "Running file \"$name\" as text filter ..."
  773.         
  774.         perlTextFilter $script
  775.     } else {
  776.         set perlPrevScript {}
  777.         synchroniseModeVar perlLastFilter $perlPrevScript 
  778.         enableMenuItem $perlMenu repeatLastFilter 0
  779.         
  780.         alertnote "Couldn't read the script file : $path"
  781.         return
  782.     }
  783. }
  784.  
  785. #############################################################################
  786. # Prepare the contents of a text window for use as a text-filter script. 
  787. # (calls perlTextFilter to actually run the script)
  788. proc perlBufferAsFilter {wname} {
  789.     global scriptFile scriptStart perlPrevScript perlMenu perlName
  790.  
  791.     set ok [regexp {(.*):([^:]*)} $wname pathname dirname name]
  792.     if {!$ok} {    set name $wname    }
  793.     
  794.     if {[lsearch [winNames -f] $wname] >= 0} {
  795.         set coreScript [getText -w $wname 0 [maxPos -w $wname]]
  796.         
  797.         # Does it have any text in it?
  798.         if {[string length $coreScript]} {
  799.             set scriptFile $wname
  800.             set scriptStart 1
  801.             set script [wrapFilterScript $coreScript]
  802.             set perlPrevScript [list "buffer" $wname]
  803.             synchroniseModeVar perlLastFilter $perlPrevScript 
  804.             enableMenuItem $perlMenu repeatLastFilter 1
  805.             message "Running buffer \"$name\" as text filter ..."
  806.             
  807.             perlTextFilter $script
  808.         }
  809.     } else {
  810.         set perlPrevScript {}
  811.         synchroniseModeVar perlLastFilter $perlPrevScript 
  812.         enableMenuItem $perlMenu repeatLastFilter 0
  813.  
  814.         alertnote "Couldn't find buffer : $name"
  815.     }
  816. }
  817.  
  818. #############################################################################
  819. # Run a Perl script as a command-line text filter, arranging for a text
  820. # buffer to be attached as standard input.  The calling routine should already
  821. # have processed the script with wrapFilterScript.  This routine actually
  822. # send the script and takes care of writing the input and reading the output 
  823. # files.
  824. proc perlTextFilter {script {args {}} {flags {}}} {
  825.     global perloverwriteSelection perlapplyToBuffer perlpromptForArgs
  826.     global filterHeadLen scriptFile scriptStart perluseDebugger ALPHA
  827.     global perlOutputWindow perlRecycleOutput perlName
  828.  
  829.     set perlName [file tail [app::launchBack McPL]]
  830.     if {![string length $perlName]} {
  831.         alertnote "Couldn't run MacPerl"
  832.         error "Couldn't run MacPerl"
  833.     }
  834.     writeStdin
  835.  
  836.     if {$perluseDebugger} {
  837.         append flags "debug"
  838.     }
  839.     if {$perlpromptForArgs} { 
  840.         append args " [getCmdlineArgs]"
  841.     }
  842.     
  843.     sendCloseWinName $perlName $perlName
  844.     sendCloseWinName $perlName "Perl Debug"
  845.     
  846.     if {$perluseDebugger} {
  847.         switchTo $perlName
  848.         perlDoScript $perlName [scriptPath] $args [list [stdinPath]] $flags
  849.         set err [getMacPerlError]
  850.  
  851.     } else {
  852.         watchCursor
  853.         set reply [perlDoScriptBatch $perlName [scriptPath] $args [list [stdinPath]]]
  854.         set err [getBatchError $reply]
  855.     }
  856.     
  857.     switchTo $ALPHA
  858.     
  859.     if {$err == 0} {
  860.         if {$perluseDebugger} {
  861.             set outp [sendGetText $perlName $perlName]
  862.         } else {
  863. #            set outp [parseReplyOutp $reply]
  864.             set outp [parseReplyResult $reply]
  865.         }
  866.         pasteFilterResult $outp
  867.     }
  868. }
  869.  
  870.  
  871. #############################################################################
  872. # Check the MacPerl output window for error messages.
  873. #
  874. proc getMacPerlError {} {
  875.     
  876.     set diag [getPerlDiag 40]
  877.     set errf [parseDiagErrf $diag]
  878.     set srcs [parseDiagSrcs $diag]
  879.     set mesg [parseDiagMesg $diag]
  880.  
  881.     if {[string length $errf]} {
  882.         showPerlDiag $diag [string length $diag] $mesg $errf $srcs
  883.         gotoPerlError $errf $srcs $mesg
  884.         return 1
  885.         
  886.     } else {
  887.         return 0
  888.     }
  889. }
  890.  
  891. #############################################################################
  892. # Check the MacPerl batch reply for error messages.
  893. #
  894. proc getBatchError {reply} {
  895.     global perlErrorWindow
  896.     
  897.     set fatalError 0
  898.     set diag [parseReplyDiag $reply]
  899.     set errf [parseDiagErrf  $diag ]
  900.     set srcs [parseReplySrcs $reply]
  901.     set mesg [parseDiagMesg  $diag ]
  902.     set errn [parseReplyErrn $reply]
  903.  
  904.     if {$errn} {        
  905.         showPerlDiag $diag $errn $mesg $errf $srcs
  906.         gotoPerlError $errf $srcs $mesg
  907.         set fatalError 1
  908.         
  909.     } elseif {[string length $diag] > 0} {
  910.         showPerlDiag $diag $errn $mesg $errf $srcs
  911.     }
  912.     
  913.     return $fatalError
  914. }
  915.  
  916. #############################################################################
  917. # Display the Perl diagnostic output in its own window.
  918. #
  919. proc showPerlDiag {diag {errn 1} {mesg {}} {errf {}} {srcs {}}} {
  920.         global perlErrorWindow    
  921.         
  922.         set currWin [lindex [winNames] 0]
  923.         if {[lsearch [winNames] $perlErrorWindow] >= 0} {
  924.             bringToFront $perlErrorWindow
  925.             setWinInfo read-only 0
  926.             deleteText 0 [maxPos] 
  927.             insertText $diag
  928.         } else {
  929.             new -n $perlErrorWindow 
  930.              insertText $diag
  931.         }
  932.         
  933.         goto 0
  934.         catch {shrinkWindow 2}
  935.         setWinInfo dirty 0
  936.         setWinInfo read-only 1
  937.         bringToFront $currWin
  938. }
  939.  
  940. #############################################################################
  941. # Bring up a window containing the bug-ridden Perl code and highlight the
  942. # line at which the error was found.
  943. #
  944. proc gotoPerlError {errf srcs {mesg {}}} {
  945.     global scriptFile scriptStart filterHeadLen
  946.  
  947.     if {$errf == [scriptPath] || $errf == "<AppleEvent>"} {
  948.         set errf $scriptFile
  949.         # Convert it to the line number in the original file
  950.         set srcs [expr $srcs + $scriptStart - $filterHeadLen - 1]
  951.     }
  952.     # ... and leave an informative error message
  953.     #
  954.     if {[string length $mesg]} {
  955.         set mesg "$mesg at Line $srcs"            
  956.     } else {
  957.         set mesg "MacPerl flagged an error at Line $srcs"    
  958.     }
  959.     
  960.     # Bring up the script file and highlight the flagged line
  961.     #
  962.     catch {gotoFileLine $errf $srcs $mesg} fname    
  963. }
  964.  
  965. #############################################################################
  966. # Read the first block of lines (up to a maximum number) from the MacPerl
  967. # output window.
  968. #
  969. proc getPerlDiag {maxlines} {
  970.     global perlName
  971.     set pat0 {^[ \t]*$}
  972.  
  973.     set lines {}    
  974.  
  975.     # read first $maxlines of output to the MacPerl window
  976.     # (faster, but assumes error message won't appear at 
  977.     # the end of a lot of output).
  978.     #
  979.     set nlines [sendCountLines $perlName MacPerl]
  980.     set nlines [expr ($nlines > $maxlines)?$maxlines:$nlines]
  981.     if {$nlines > 0} {
  982.         set output [sendGetText $perlName $perlName 1 $nlines]
  983.         
  984.         foreach line [split $output "\r"] {
  985.             if  {[regexp $pat0 $line mtch]} {
  986.                 break
  987.             } else {
  988.                 append lines "$line\n"
  989.             }
  990.         }
  991.     }
  992.     return $lines
  993. }
  994.  
  995. #############################################################################
  996. # Extract various items out of the MacPerl diagnostic output
  997. #
  998.  
  999. # Name of the file in which the error was found
  1000. #
  1001. proc parseDiagErrf {diag}    {
  1002.     if {![regexp {File '([^']+)'; Line} $diag allofit errf]} { 
  1003.         set errf {}
  1004.     }
  1005.     return $errf
  1006. }
  1007.  
  1008. # The line number on which the error was found
  1009. #
  1010. proc parseDiagSrcs {diag}    {
  1011.     if {![regexp {File '[^']+'; Line ([0-9]+)} $diag allofit srcs]} { 
  1012.         set srcs 0 
  1013.     }
  1014.     return $srcs
  1015. }
  1016.  
  1017. # The error message associated with error
  1018. #
  1019. proc parseDiagMesg {diag} {
  1020.     set pat1 {^#(.*)$}
  1021.     set pat2 {File '([^']+)'; Line ([0-9]+)}
  1022.     
  1023.     set errMessage {}
  1024.     set errFound 0
  1025.     
  1026.     foreach line [split $diag "\n"] {
  1027.         if {[regexp $pat2 $line mtch num]} {
  1028.             set errFound 1
  1029.         } elseif {[regexp $pat1 $line mtch err]} {
  1030.             if {$errFound == 0} {
  1031.                 set errMessage $err
  1032.             }
  1033.         }
  1034.     }
  1035.     return $errMessage
  1036. }
  1037.  
  1038. #############################################################################
  1039. # Extract various return parameters out of a MacPerl DoScript reply
  1040. #
  1041.  
  1042. # Result from batch script
  1043. #
  1044. proc parseReplyResult {reply} {
  1045.     if {![regexp {'?\-\-\-\-'?:“([^”]*)”} $reply allofit result]} { 
  1046.         set result {}
  1047.     }
  1048.     return $result
  1049. }
  1050.  
  1051. # Standard output of batch script
  1052. #
  1053. proc parseReplyOutp {reply} {
  1054.     if {![regexp {OUTP:“([^”]*)”} $reply allofit outp]} { 
  1055.         set outp {}
  1056.     }
  1057.     return $outp
  1058. }
  1059.  
  1060. # Diagnostic output of the batch script
  1061. #
  1062. proc parseReplyDiag {reply}    {
  1063.     if {[regexp {diag:“([^”]*)”} $reply allofit diag]}  {
  1064.     } else { 
  1065.         set diag {}
  1066.     }
  1067.     return $diag
  1068. }
  1069.  
  1070. # File alias of the script file in which the error was found
  1071. #
  1072. proc parseReplyErob {reply}    {
  1073.     if {![regexp {erob:alis\(«(.*)»\)} $reply allofit erob]} {
  1074.         set erob {} 
  1075.     }
  1076.     return $erob
  1077. }
  1078.  
  1079. # First line flagged in error
  1080. #
  1081. proc parseReplySrcs {reply}    {
  1082.     if {![regexp {erng:{srcs:([0-9]+)[^\}]*}} $reply allofit srcs]} { 
  1083.         set srcs 0 
  1084.     }
  1085.     return $srcs
  1086. }
  1087.  
  1088. # Last line flagged in error
  1089. #
  1090. proc parseReplySrce {reply}    {
  1091.     if {![regexp {erng:{[^\}]*srce:([0-9]+)}} $reply allofit srce]} { 
  1092.         set srce 0
  1093.     }
  1094.     return $srce
  1095. }
  1096.  
  1097. # Error number
  1098. #
  1099. proc parseReplyErrn {reply}    {
  1100.     if {![regexp {errn:([0-9]+)} $reply allofit errn]} {
  1101.         set errn 0
  1102.     }
  1103.     return $errn
  1104. }
  1105.  
  1106. #############################################################################
  1107. #  Take a Perl script and add commands to take the file STDIN as standard
  1108. #  input and STDOUT as standard output.  This allows scripts written as
  1109. #  Unix command-line filters to be used in the (non-MPW) Mac environment as
  1110. #  text filters.
  1111. #
  1112. #  If there's already a #! line in the script, then the new commands
  1113. #  are added after that line.  If there was no #! line in the first place,
  1114. #  one is added, in case MacPerl is set up to require it (can't hurt...) 
  1115. #
  1116. #  $filterHeadLen counts the number of lines we add to the top of the
  1117. #  original script, so that we can allow for it in interpreting error
  1118. #  messages issued by MacPerl.
  1119. #
  1120. #  *** As of MacPerl 4.1.4, this business is pretty much obsolete ***
  1121. #
  1122. proc wrapFilterScript {coreScript} {
  1123.     global scriptStart filterHeadLen interpPat
  1124.  
  1125.     if {[regexp -indices $interpPat $coreScript allofit cmdln]} {
  1126.         set endPos [lindex $cmdln 1]
  1127.         set filterHead [string range $coreScript 0 [expr $endPos+1]]
  1128.         set coreScript [string range $coreScript [expr $endPos+2] end]
  1129.         set filterHeadLen 0
  1130.         incr scriptStart [expr [llength [split $filterHead "\n\r"]] -2]
  1131.     } else {
  1132.         set filterHead "#!/bin/perl\r\n"
  1133.         set filterHeadLen 2
  1134.     }
  1135.         
  1136.     set script $filterHead
  1137.     append script $coreScript
  1138.     
  1139.     # for debugging purposes, save the script on disk
  1140.     #
  1141.     writeScript $script
  1142.     return $script
  1143. }        
  1144.  
  1145. #############################################################################
  1146. #  Add a #!/bin/perl line to the script if it doesn't contain one already.
  1147. #  (MacPerl puts up dialog if this line is missing when it expects it,
  1148. #  hanging the DoScript and leaving us stuck.)
  1149. #
  1150. proc wrapSelectScript {coreScript} {
  1151.     global scriptStart filterHeadLen interpPat
  1152.  
  1153.     if {[regexp -indices $interpPat $coreScript allofit cmdln]} {
  1154.         set endPos [lindex $cmdln 1]
  1155.         set filterHead [string range $coreScript 0 [expr $endPos+1]]
  1156.         set script $coreScript
  1157.         set filterHeadLen 0
  1158.         incr scriptStart [expr [llength [split $filterHead "\n\r"]] -2]
  1159.     } else {
  1160.         set script "#!/bin/perl\r\n"
  1161.         append script $coreScript
  1162.         set filterHeadLen 1
  1163.     }
  1164.     
  1165.     # for debugging purposes, save the script on disk
  1166.     #
  1167.     writeScript $script
  1168.     return $script
  1169. }        
  1170.  
  1171. #############################################################################
  1172. #  Paste result of the filter operation in place of the input text, or in
  1173. #  a new window (depending on the flag $perloverwriteSelection
  1174. #
  1175. proc pasteFilterResult {text} {
  1176.     global perloverwriteSelection perlRecycleOutput perlOutputWindow
  1177.     global perlapplyToBuffer 
  1178.     
  1179.     if {!$perloverwriteSelection} {
  1180.         if {$perlRecycleOutput && 
  1181.             [lsearch [winNames] $perlOutputWindow] >= 0} {                
  1182.             bringToFront $perlOutputWindow
  1183.         } else {
  1184.             new -n $perlOutputWindow
  1185.         }
  1186.     }
  1187.     
  1188.     if {$perlapplyToBuffer || $perlRecycleOutput} {
  1189.         set from 0
  1190.         set to [maxPos]
  1191.     } else {
  1192.         set from [getPos] 
  1193.         set to [selEnd]
  1194.     }    
  1195.     replaceText $from $to $text
  1196.     
  1197.     if {!$perloverwriteSelection || $perlapplyToBuffer} {
  1198.         catch {shrinkWindow 2}
  1199.         goto 0
  1200.     } else {
  1201.         catch shrinkWindow
  1202.         goto $from
  1203.     }
  1204.     if {!$perloverwriteSelection} { setWinInfo dirty 0 }
  1205. }    
  1206.  
  1207. #############################################################################
  1208. #  Extend the current selection to encompass complete lines.  If the 
  1209. #  'applyToBuffer' flag is checked, then the entire buffer is selected.
  1210. #
  1211. proc completeSelection {} {
  1212.     global perlapplyToBuffer filterInput
  1213.     set filterInput "buffer \"[lindex [winNames] 0]\""
  1214.     if {$perlapplyToBuffer} {
  1215.         set start 0
  1216.         set end [maxPos]
  1217.     } else {
  1218.         set start [lineStart [getPos]]
  1219.         set end [nextLineStart [expr [selEnd]-1]]
  1220.         if {$end == $start} { set end [nextLineStart [selEnd]] }
  1221.         
  1222.         set startLine [lindex [posToRowCol $start] 0]
  1223.         set endLine [expr [lindex [posToRowCol $end] 0] - 1]
  1224.         if {$endLine > $startLine+1} {
  1225.             set filterInput "lines $startLine to $endLine of $filterInput"
  1226.         } else {
  1227.             set filterInput "line $startLine of $filterInput"
  1228.         }
  1229.    }
  1230.     return [list $start $end]
  1231. }
  1232.  
  1233. #############################################################################
  1234. #  writeStdin: Extend the selection, as appropriate, and write it to the 
  1235. #     STDIN file in the MacPerl directory.
  1236. #
  1237. #  writeScript: Write the SCRIPT file in the MacPerl directory.  MacPerl will
  1238. #     read the script from this file. 
  1239. #
  1240. proc writeStdin {} {
  1241.     set res [completeSelection]
  1242.     set tmpfid [open [stdinPath] "w+"]
  1243.     puts $tmpfid [eval getText $res]
  1244.     close $tmpfid
  1245. }
  1246.  
  1247. # This is unnecessary now, but maybe it'll still useful to save the script
  1248. # file for debugging.
  1249. #
  1250. proc writeScript {script} {
  1251.     set tmpfid [open [scriptPath] "w+"]
  1252.     puts $tmpfid $script 
  1253.     close $tmpfid
  1254. }
  1255.  
  1256. #############################################################################
  1257. # Read the MacPerl output window and load the contents, if any, into
  1258. # a new Alpha window. 
  1259. #
  1260. proc openPerlOutput {} {
  1261.     global perlRecycleOutput perlOutputWindow perlName
  1262.     
  1263.     set output [sendGetText $perlName $perlName]
  1264.     if {[string length $output]} {
  1265.         if {$perlRecycleOutput && 
  1266.             [lsearch [winNames] $perlOutputWindow] >= 0} {
  1267.             
  1268.             bringToFront $perlOutputWindow
  1269.             replaceText 0 [maxPos] $output
  1270.         } else {
  1271.             new -n $perlOutputWindow
  1272.             insertText $output
  1273.         }
  1274.         catch {shrinkWindow 2}
  1275.         setWinInfo dirty 0
  1276.         goto 0
  1277.     }
  1278. }
  1279.  
  1280. #############################################################################
  1281. # translate special DoScript flags into flags string $usrf
  1282. #
  1283. proc perlScriptFlags {{flags {}}} {
  1284.      set usrf {}
  1285.  
  1286.     if {[lsearch -exact $flags "extract"] >= 0} {
  1287.         append usrf { "EXTR" 'true'}
  1288.     } elseif {[lsearch -exact $flags "noextract"] >= 0} {
  1289.         append usrf { "EXTR" 'fals'}
  1290.     }        
  1291.     if {[lsearch -exact $flags "debug"] >= 0} {
  1292.         append usrf { "DEBG" 'true'}
  1293.     } elseif {[lsearch -exact $flags "nodebug"] >= 0} {
  1294.         append usrf { "DEBG" 'fals'}
  1295.     }        
  1296.  
  1297.     if {[lsearch -exact $flags "local"] >= 0} {
  1298.         append usrf { "MODE" 'LOCL'}
  1299.     } elseif {[lsearch -exact $flags "batch"] >= 0} {
  1300.         append usrf { "MODE" 'BATC'}
  1301.     } elseif {[lsearch -exact $flags "remote"] >= 0} {
  1302.         append usrf { "MODE" 'RCTL'}
  1303.     }        
  1304.     return $usrf
  1305.  
  1306. proc perlScriptArgs {{args {}} {fileargs {}}} {
  1307.     set nargs 0
  1308.     set argv {}
  1309.     
  1310.     foreach item [parseWords $args] {
  1311.         set item [string trim $item]
  1312.         if {[string length $item]} {
  1313.             append argv ", [curlyq $item]"
  1314.             incr nargs
  1315.         }
  1316.     }
  1317.     foreach filename $fileargs {
  1318.         set item [string trim $filename]
  1319.         if {[string length $item]} {
  1320.             append argv ", [curlyq $item]"
  1321.             incr nargs
  1322.         }
  1323.     }
  1324.     return $argv
  1325. }
  1326.  
  1327. #############################################################################
  1328. # General Apple Event routines
  1329. # (most of these have been moved to Modes:appleEvents.tcl)
  1330. #
  1331. # DoScript for MacPerl 4.1.3
  1332. # (runs in "Local" mode under v4.1.4+)
  1333. #
  1334. proc perlDoScript {appname script {args {}} {fileargs {}} {flags {}} } {
  1335.     # form list of quoted "command-line" args
  1336.     #
  1337.     if {$script != ""} {
  1338.         set argv "\[[curlyq [string trim $script]]"
  1339.         append argv [perlScriptArgs $args $fileargs]
  1340.         append argv "]"
  1341.         
  1342.         set usrf [perlScriptFlags $flags]
  1343.         set reply [eval "AEBuild -t 36000 -r \"$appname\" misc dosc $usrf \"----\" [list $argv] "]
  1344.     #    alertnote $reply
  1345.     }
  1346. }
  1347.  
  1348. # DoScript for MacPerl 4.1.4+
  1349. #
  1350. proc perlDoScriptBatch {appname script {args {}} {fileargs {}}} {
  1351.     
  1352.     # form list of quoted "command-line" args
  1353.     #
  1354.     if {$script != ""} {
  1355.         set argv "\[[curlyq [string trim $script]]"
  1356.         append argv [perlScriptArgs $args $fileargs ] 
  1357.         append argv "]"
  1358.                 
  1359.         set reply [eval "AEBuild -t 36000 -r \"$appname\" misc dosc MODE BATC \"----\" [list $argv]"]
  1360.         
  1361. #         perlDisplayReply $reply
  1362.  
  1363.     } else {
  1364.         set reply {}
  1365.     }
  1366.     return $reply
  1367. }
  1368.  
  1369. # For debugging 
  1370. #
  1371. proc perlDisplayReply {reply} {
  1372.     set currWin [lindex [winNames] 0]
  1373.     new -n {*** DoScript Reply **} 
  1374.     insertText $reply
  1375.         
  1376.     goto 0
  1377.     catch {shrinkWindow 2}
  1378.     setWinInfo dirty 0
  1379.     setWinInfo read-only 1
  1380.     bringToFront $currWin
  1381. }
  1382.  
  1383. # DoScript to launch interactive debugger (for MacPerl 4.1.4+)
  1384. #
  1385. proc perlDoScriptDebug {appname script {args {}} {fileargs {}}} {
  1386.     
  1387.     # form list of quoted "command-line" args
  1388.     #
  1389.     if {$script != ""} {
  1390.         set argv "\[[curlyq [string trim $script]]"
  1391.         append argv [perlScriptArgs "$args debug" $fileargs ] 
  1392.         append argv "]"
  1393.                 
  1394.         set reply [eval "AEBuild -t 36000 -r \"$appname\" misc dosc MODE RCTL \"----\" [list $argv]"]
  1395.  
  1396.         new -n {** DoScriptDebug Reply **} 
  1397.         insertText $reply
  1398.             
  1399.         goto 0
  1400.         catch {shrinkWindow 2}
  1401.         setWinInfo dirty 0
  1402.         setWinInfo read-only 1
  1403.  
  1404.  
  1405.     } else {
  1406.         set reply {}
  1407.     }
  1408.     return $reply
  1409. }
  1410.  
  1411. ##############################################################################
  1412. # Automatic indexing of Perl subs
  1413. #
  1414. proc Perl::MarkFile {} {
  1415.     set end [maxPos]
  1416.     set pos 0
  1417.     set l {}
  1418.     while {![catch {search -f 1 -r 1 -m 0 -i 0 {^sub} $pos} res]} {
  1419.         set start [lindex $res 0]
  1420.         set end [nextLineStart $start]
  1421.         set text [lindex [getText $start $end] 1]
  1422.         set pos $end
  1423.         set inds($text) [lineStart [expr $start - 1]]
  1424.     }
  1425.  
  1426.     if {[info exists inds]} {
  1427.         foreach f [lsort [array names inds]] {
  1428.             set next [nextLineStart $inds($f)]
  1429.             setNamedMark $f $inds($f) $next $next
  1430.         }
  1431.     }
  1432. }
  1433.  
  1434.  
  1435. # Open a 'require'd Perl file.
  1436. proc perlFindRequire {from {to 0}} {
  1437.     set reqPat {^[     ]*require[     ]*(\"[^\"]+\"|\'[^\']+\'|[^     ]+)}
  1438.     if {$to == 0} { set to $from }
  1439.     set beg [lineStart $from]
  1440.     set end [nextLineStart $to]
  1441.     set words [parseWords [getText $beg $end]]
  1442.     if {[string tolower [lindex $words 0]] != "require"} {
  1443.         error "Not a require statement"
  1444.     }
  1445.     set root [string trim [lindex $words 1] {'"}]
  1446.     return $root
  1447. }
  1448.  
  1449. proc inlineRequires {} {
  1450.     global lastMatchingLines
  1451.     
  1452.     set reqPat {^[     ]*require[     ]*(\"[^\"]+\"|\'[^\']+\'|[^     ]+)}
  1453.     set pos 0
  1454.     while {![catch {search -s -f 1 -r 1 -m 0 -i 1 $reqPat $pos} mtch]} {
  1455.          [lindex [posToRowCol [lindex $mtch 0]] 0]] 
  1456.         set name [string [eval getText $mtch]
  1457.         set pos [lindex $mtch 1]
  1458.         incr matches
  1459.     }
  1460. }
  1461.  
  1462. # Open a Perl source file. 
  1463. #
  1464. proc openPerlFile {file {extensions {""}}} {
  1465.     global perlSearchPath
  1466.     # Determine absolute file specification
  1467.     # Ignore $extensions if $file already has an extension
  1468.     if {[string length [file extension $file]] == 0} {
  1469.         set extensions {""}
  1470.     }
  1471.     foreach ext $extensions {
  1472.         set filename [absolutePath $file$ext]
  1473.         if {![catch {openFileQuietly $filename}]} {
  1474.             message $filename
  1475.             return 
  1476.         }
  1477.     }
  1478.     if {[llength $perlSearchPath] == 0} { buildPerlSearchPath }
  1479.     foreach folder $perlSearchPath {
  1480.         foreach ext $extensions {
  1481.             set filename "$folder$file$ext"
  1482.             if {![catch {openFileQuietly $filename}]} {
  1483.                 message $filename
  1484.                 return     
  1485.             }
  1486.         }
  1487.     }
  1488.     beep
  1489.     message "can't find Perl source file \"$file\""
  1490. }
  1491.  
  1492. # Return a list of folders in which to search for Perl library files, 
  1493. # including the lib folder in the Perl application directory and the
  1494. # $perlLibFolder folder (if it exists) .  
  1495. # The current folder is not included in the list.
  1496. #
  1497. # (The $perlLibFolder folder is assigned from the AppPaths submenu.)
  1498. #
  1499. proc buildPerlSearchPath {} {
  1500.     global perlLibFolder perlSearchPath
  1501.     message "building Perl search path..."
  1502.     set folders {}
  1503.     
  1504.     # The local lib folder:
  1505.     if {[info exists perlLibFolder] && [string length $perlLibFolder] > 0} { 
  1506.         set folders [concat $folders [list $perlLibFolder]]
  1507.         # Search subfolders one level deep:
  1508.         set folders [concat $folders [listSubfolders $perlLibFolder 1]]
  1509.     }
  1510.  
  1511.     # Any "*lib*" folders in the MacPerl application folder:
  1512.     set macperlPath [nameFromAppl McPL]
  1513.     set appDir [file dirname $macperlPath]
  1514.     set folders [concat $folders [list $appDir]]
  1515.     # Bug:  'glob' is case sensitive!
  1516.     foreach folder [glob "$appDir:*\[Ll\]ib*"] {
  1517.         set folders [concat $folders [list $folder]]
  1518.         # Search subfolders one level deep:
  1519.         set folders [concat $folders [listSubfolders $folder 1]]
  1520.     }
  1521.  
  1522.     # Make sure each folder ends with a colon
  1523.     set perlSearchPath {}
  1524.     foreach folder $folders {
  1525.         set folder "[string trimright $folder {:}]:"
  1526.         set perlSearchPath [concat $perlSearchPath [list $folder]]
  1527.     }
  1528. }
  1529.  
  1530. ###########################################################################
  1531. #
  1532. proc perlHelpProc {menu item} {
  1533.     global HOME
  1534.     switch $item {
  1535.         "MacPerl Mode"    {
  1536.                 if {[catch {openFileQuietly "$HOME:Help:MacPerl Help"}]} {
  1537.                     alertnote "File not found:\r$HOME:Help:MacPerl Help"
  1538.                 }
  1539.             }
  1540.         "Mac Specifics"    {
  1541.                 if {[catch {openFileQuietly "$HOME:Help:MacPerl.Specifics"}]} {
  1542.                     alertnote "File not found:\r$HOME:Help:MacPerl.Specifics"
  1543.                 }
  1544.             }
  1545.         "Perl4 Manual"    {
  1546.                 if {[catch {openFileQuietly "$HOME:Help:Perl Commands"}]} {
  1547.                     alertnote "File not found:\r$HOME:Help:Perl Commands"
  1548.                 }
  1549.             }
  1550.         "Perl5 Manual"    {
  1551.                 catch {editMark "$HOME:Help:Perl Commands" Perl5 -r}
  1552.             }
  1553.     }
  1554. }
  1555.  
  1556. proc Perl::electricLeft {} {
  1557.     set prevChar [lookAt [expr [getPos] - 1]]
  1558.     if {$prevChar == " " || $prevChar == "\)"} {
  1559.         # Trick to continue with the generic function.
  1560.         error "Use generic function!"
  1561.     }
  1562.     deleteText [getPos] [selEnd]
  1563.     insertText "\{"
  1564. }
  1565.  
  1566. proc Perl::electricRight {} {
  1567.     set prevChar [lookAt [expr [getPos] - 1]]
  1568.     if {$prevChar == " " || $prevChar == ";" || $prevChar == "\t" || $prevChar == "\}"} {
  1569.         # Trick to continue with the generic function.
  1570.         error "Use generic function!"
  1571.     }
  1572.     deleteText [getPos] [selEnd]
  1573.     insertText "\}"
  1574.     catch {blink [matchIt "\}" [expr [getPos]-2]]}
  1575.     return
  1576. }
  1577.  
  1578.  
  1579.